1   /*
2    * Copyright 2002-2019 the original author or authors.
3    *
4    * Licensed under the Apache License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    *      https://www.apache.org/licenses/LICENSE-2.0
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  
17  package org.springframework.integration.transformer;
18  
19  import java.io.IOException;
20  import java.io.UncheckedIOException;
21  import java.util.Collection;
22  import java.util.HashMap;
23  import java.util.Map;
24  import java.util.Map.Entry;
25  
26  import org.springframework.integration.support.json.JsonObjectMapper;
27  import org.springframework.integration.support.json.JsonObjectMapperProvider;
28  import org.springframework.util.Assert;
29  import org.springframework.util.CollectionUtils;
30  import org.springframework.util.StringUtils;
31  
32  /**
33   * Transforms an object graph into a Map. It supports a conventional Map (map of maps)
34   * where complex attributes are represented as Map values as well as a flat Map
35   * where keys document the path to the value. By default it will transform to a flat Map.
36   * If you need to transform to a Map of Maps set the 'shouldFlattenKeys' property to 'false'
37   * via the {@link ObjectToMapTransformer#setShouldFlattenKeys(boolean)} method.
38   * It supports Collections, Maps and Arrays which means that for flat maps it will flatten
39   * an Object's properties. Below is an example showing how a flattened
40   * Object hierarchy is represented when 'shouldFlattenKeys' is TRUE.
41   *<p>
42   * The transformation is based on to and then from JSON conversion.
43   *
44   * <code>
45   * public class Person {
46   *     public String name = "John";
47   *     public Address address = new Address();
48   * }
49   * public class Address {
50   *     private String street = "123 Main Street";
51   * }
52   * </code>
53   *
54   * The resulting Map would look like this:
55   * <code>
56   * {name=John, address.street=123 Main Street}
57   * </code>
58   *
59   * @author Oleg Zhurakousky
60   * @author Artem Bilan
61   * @author Gary Russell
62   * @author Vikas Prasad
63   *
64   * @since 2.0
65   *
66   * @see JsonObjectMapperProvider
67   */
68  public class ObjectToMapTransformer extends AbstractPayloadTransformer<Object, Map<?, ?>> {
69  
70  	private final JsonObjectMapper<?, ?> jsonObjectMapper;
71  
72  	private volatile boolean shouldFlattenKeys = true;
73  
74  	/**
75  	 * Construct with the default {@link JsonObjectMapper} instance available via
76  	 * {@link JsonObjectMapperProvider#newInstance() factory}.
77  	 */
78  	public ObjectToMapTransformer() {
79  		this(JsonObjectMapperProvider.newInstance());
80  	}
81  
82  	/**
83  	 * Construct with the provided {@link JsonObjectMapper} instance.
84  	 * @param jsonObjectMapper the {@link JsonObjectMapper} to use.
85  	 * @since 5.0
86  	 */
87  	public ObjectToMapTransformer(JsonObjectMapper<?, ?> jsonObjectMapper) {
88  		Assert.notNull(jsonObjectMapper, "'jsonObjectMapper' must not be null");
89  		this.jsonObjectMapper = jsonObjectMapper;
90  	}
91  
92  	public void setShouldFlattenKeys(boolean shouldFlattenKeys) {
93  		this.shouldFlattenKeys = shouldFlattenKeys;
94  	}
95  
96  	@Override
97  	@SuppressWarnings("unchecked")
98  	protected Map<String, Object> transformPayload(Object payload) {
99  		Map<String, Object> result;
100 		try {
101 			result = this.jsonObjectMapper.fromJson(this.jsonObjectMapper.toJson(payload), Map.class);
102 		}
103 		catch (IOException e) {
104 			throw new UncheckedIOException(e);
105 		}
106 		if (this.shouldFlattenKeys) {
107 			result = this.flattenMap(result);
108 		}
109 		return result;
110 	}
111 
112 	@Override
113 	public String getComponentType() {
114 		return "object-to-map-transformer";
115 	}
116 
117 	@SuppressWarnings("unchecked")
118 	private void doProcessElement(String propertyPrefix, Object element, Map<String, Object> resultMap) {
119 		if (element instanceof Map) {
120 			this.doFlatten(propertyPrefix, (Map<String, Object>) element, resultMap);
121 		}
122 		else if (element instanceof Collection) {
123 			this.doProcessCollection(propertyPrefix, (Collection<?>) element, resultMap);
124 		}
125 		else if (element != null && element.getClass().isArray()) {
126 			Collection<?> collection = CollectionUtils.arrayToList(element);
127 			this.doProcessCollection(propertyPrefix, collection, resultMap);
128 		}
129 		else {
130 			resultMap.put(propertyPrefix, element);
131 		}
132 	}
133 
134 	private Map<String, Object> flattenMap(Map<String, Object> result) {
135 		Map<String, Object> resultMap = new HashMap<>();
136 		this.doFlatten("", result, resultMap);
137 		return resultMap;
138 	}
139 
140 	private void doFlatten(String propertyPrefixArg, Map<String, Object> inputMap, Map<String, Object> resultMap) {
141 		String propertyPrefix = propertyPrefixArg;
142 		if (StringUtils.hasText(propertyPrefix)) {
143 			propertyPrefix = propertyPrefix + ".";
144 		}
145 		for (Entry<String, Object> entry : inputMap.entrySet()) {
146 			this.doProcessElement(propertyPrefix + entry.getKey(), entry.getValue(), resultMap);
147 		}
148 	}
149 
150 	private void doProcessCollection(String propertyPrefix, Collection<?> list, Map<String, Object> resultMap) {
151 		int counter = 0;
152 		for (Object element : list) {
153 			this.doProcessElement(propertyPrefix + "[" + counter + "]", element, resultMap);
154 			counter++;
155 		}
156 	}
157 
158 }